home *** CD-ROM | disk | FTP | other *** search
Wrap
# # NEWS v0.4 # eggdrop/tcl script by eden <brookes@cs.uq.edu.au> # This script adds commands so that users can post news for others # to read. It works similarly to the notes system, except news can be # read by anybody. Users can add news messages, erase news messages, view # them privately (by NOTICE) or display them to the channel. A timer is # included that alerts users on channel to the presence of notes. # # It is based on the original news script by Robey, and the timer stuff was # based on the tcl.topic script by Brian_C <s0206377@cc.ysu.edu>. # # # /msg <bot> news add <message...> # (requires to be known by the bot) # adds a message to the news file on the bot # # /msg <bot> news erase # # (requires to be known by the bot) # erases the message numbered # from the news file. Only the user who # put the message there may erase it, except for masters who can erase # any message # # /msg <bot> news read \[#\] # (anyone can do) # shows the current news file, via notice. The item number # is # optional. If there, only that item is shown. # # /msg <bot> news show \[#\] # (anyone can do) # shows the current news file publicly, to the channel. Will not show # the newsfile more than once a minute, to prevent flooding. The item # number is optional. If there, only that item is shown. # # /msg <bot> news announce # (anyone can do) # announces the presence of news to the channel (if there is any) # # /msg <bot> news check # (anyone can do) # sends a private notice to a user telling them how much news there is # # /msg <bot> news reset # (requires +m) # resets the counter associated with the news system but does not # modify the news file itself. Shouldn't be necessary in an ideal world :) # # /msg <bot> news clear # (requires +m) # erases all news in the news file and resets the counter # # There is also a timer included, so that once an hour, on the hour, the # bot displays an announcement that there is news to be read (if there is). # It doesn't display the news to the channel unless asked. # # All TCL variables and internal procedures associated with the news system # start with "news" # # TODO: # - DCC bindings so the news can be manipulated on the party line # # HISTORY: # - 0.3 -> 0.4 # Changed basic way of informing users of news. Now it is done with an # on_join binding. It keeps a record for EACH user of the timestamp of # the last message they read, and only shows them unread messages by # default. This info is stored in the user's "comment" field. # Added "check" message command. # # - 0.2 ->0.3 # Added "announce" message command # When users read news, it now also tells them how to get further help # Merged the internal procs news_select and news_print to just news_print # Added feedback to the "reset" command # # - 0.2 # Original version 0.2 to distinguish it from Robey's version # You may want to move this line to your config-file for easier changing #set newsfile ".news" # You can choose either to use the "umbrella" news binding, or to have # separate bindings for each command, or both. I use both because users # are dumb :) bind msg - news msg_news bind msg - addnews news_add bind msg - delnews news_del bind msg - erasenews news_del bind msg - readnews news_read bind msg - shownews news_show bind msg - announcenews news_announce bind msg - checknews news_check bind msg - resetnews news_reset bind msg - clearnews news_clear # The following line makes the script send a notice to each user as they # join the channel advising them of the unread channel news bind join - *@* news_onjoin # To DISABLE the timer, uncomment the following line. # If you have already loaded the script and it is running, you will have to # find and manually kill the timer (".tcl timers" then ".tcl killtimer #") #set set_news_running 1 #-=-=-=-=-=-=-=-=- YOU SHOULD NOT NEED TO EDIT BELOW THIS LINE -=-=-=-=-=-=-=-=- # These are internal variables and shouldn't be changed here set news_ncount -1 set news_newnews 0 set news_lastseen 0 set news_biggest_ts 0 # Is there any news? proc news_thereisnews {} { global newsfile if {[news_count] == 0} { return 0 } else { return 1 } } # How many news items are there? The first time this is called, it will # read through the file and set the value of news_ncount. In future, the # value of the variable will be used, rather than reading the file each time. # Also updates the news_biggest_ts proc news_count {} { global newsfile news_ncount news_biggest_ts # only ever go through reading the file once -- the first time it's called if {$news_ncount == -1} { set news_ncount 0 set fd [open $newsfile r] while {![eof $fd]} { set inp [gets $fd] if {[eof $fd]} {break} if {[string trim $inp " "] == ""} {continue} if {[lindex $inp 1] > $news_biggest_ts} { set news_biggest_ts [lindex $inp 1] } incr news_ncount } close $fd } return $news_ncount } # Internal procedure that, given a users handle, tells how many news items # there are in the news file that they haven't read proc news_unread {hand} { global newsfile set unread 0 set user_ts [news_get_ts $hand] set fd [open $newsfile r] while {![eof $fd]} { set inp [gets $fd] if {[eof $fd]} {break} if {[string trim $inp " "] == ""} {continue} if {[lindex $inp 1] > $user_ts} { incr unread } } close $fd return $unread } # Internal procedure to print out news from the file, since it's done more # than once. 'tochannel' should be 1 if the output is going to a channel, # or 0 if it is going privately to a user. 'item' should = 0 if all items # are to be printed, or a single number to print a specific item. # If the timestamp is non-zero, only news items more recent than this # timestamp are printed. proc news_print {tochannel nick item timestamp} { global newsfile set next 1 set fd [open $newsfile r] while {![eof $fd]} { set inp [gets $fd] if {[eof $fd]} {break} if {[string trim $inp " "] == ""} {continue} if {($item == 0) || ($item == $next)} { # timestamp == 0 to show all items if {[lindex $inp 1] > $timestamp} { set who [lindex $inp 0] set date [lrange [ctime [lindex $inp 1]] 0 2] set news [lrange $inp 2 end ] # This gets messy because there's four possible cases if {$tochannel} { if {$item == 0} { putserv [format "PRIVMSG %s :%d. \[%s\] %s \(%s\)" $nick $next \ $who $news $date] } else { putserv [format "PRIVMSG %s :\[%s\] %s \(%s\)" $nick $who \ $news $date] } } else { if {$item == 0} { puthelp [format "NOTICE %s :%d. \[%s\] %s \(%s\)" $nick $next \ $who $news $date] } else { puthelp [format "NOTICE %s :\[%s\] %s \(%s\)" $nick $who \ $news $date] } } } } incr next } close $fd return 1 } # Get a users "last read" timestamp from the userfile proc news_get_ts {hand} { set usercomment [getcomment $hand] set have_ts [scan $usercomment "%d " the_ts] if {$have_ts > 0} { return $the_ts } else { return 0 } } # Set a users "last read" timestamp in the userfile proc news_set_ts {hand new_ts} { set usercomment [getcomment $hand] set have_ts [scan $usercomment "%d" old_ts] if {$have_ts == 0} { set the_comment $usercomment } if {$have_ts == 2} { set the_comment [string range $usercomment [string first " " $usercomment] end] } set new_comment $new_ts if {($have_ts == 2) || ($have_ts == 0)} { append new_comment " " $the_comment } setcomment $hand $new_comment return } # Happens when any user joins a channel proc news_onjoin {nick uhost hand} { global botnick channel # If not known by the bot, return immediately if {[string compare $hand "*"] == 0} {return} set unread [news_unread $hand] if {$unread} { putserv "NOTICE $nick :There is $channel news ($unread unread, [news_count] total). \002/msg $botnick NEWS READ\002" } return } # Announce that there is news (if there is any) proc news_int_announce {} { global channel botnick news_newnews if {[news_thereisnews]} { if {$news_newnews} { putserv "PRIVMSG $channel :\002There is \026NEW\026 $channel news since the last hour \([news_count] items\)! To display it, /msg $botnick NEWS READ\002" putlog "NEWS: announced NEW news to channel" } else { putserv "PRIVMSG $channel :\002There is $channel news \([news_count] items\). To display it, /msg $botnick NEWS READ\002" putlog "NEWS: announced news to channel" } } else { putlog "NEWS: no news to announce" return 0 } set news_newnews 0 return 1 } # Announce the news once an hour proc news_timer {} { } # Make sure it is announced on the hour, not at any random time proc news_sync {} { set time_ending [string range [time] 2 4] if { $time_ending == ":00" } { news_timer } { timer 1 news_sync } } # Start the timer running if it's not already # NB. This is NOT a procedure, it is global. if {![info exists set_news_running]} { news_sync set set_news_running 1 } # Add a news item proc news_add {nick uhost hand arg} { global newsfile botnick news_ncount news_newnews news_biggest_ts if {![validuser $hand]} { puthelp "NOTICE $nick :Please introduce yourself to me first with \002/msg $botnick hello\002" return 0 } set message [lrange $arg 1 end] if {$message == ""} { putserv "NOTICE $nick :Usage: /msg $botnick news add <your news...>" putserv "NOTICE $nick : adds something to the news file" return 0 } set fd [open $newsfile a] puts $fd [format "%s %s %s" $hand [unixtime] $message] close $fd # Do not just use 'incr' in case news_count currently = -1 set news_ncount [expr [news_count] + 1] set news_newnews 1 set news_biggest_ts [unixtime] putserv "NOTICE $nick :Added to the news file, thanks!" return 1 } # Delete a news item proc news_del {nick uhost hand arg} { global newsfile botnick news_ncount if {![validuser $hand]} { puthelp "NOTICE $nick :Please introduce yourself to me first with \002/msg $botnick hello\002" return 0 } set otherargs [lrange $arg 1 end] if {[scan $otherargs "%d" which] != 1} { news_help $nick $uhost $hand $arg ; return 0 } if {($which < 1) || ($which > [news_count])} { puthelp "NOTICE $nick :The item number must be between 1 and [news_count]" return 0 } # We will copy the entries from the original file to a temp file, except # for the one to be deleted, then rename the temp file to be the original. # This is how eggdrop deletes notes, so we will use it too. set next 1 set fd [open $newsfile r] set newfd [open "${newsfile}~new" w] while {![eof $fd]} { set inp [gets $fd] if {[eof $fd]} {break} if {[string trim $inp " "] == ""} {continue} set who [lindex $inp 0] if {$next == $which} { # Can only delete your own news unless you're a master if {([string tolower $who] != [string tolower $hand]) && (![matchattr $hand "m"])} { puthelp "NOTICE $nick :You can\'t delete other people\'s news!" puts $newfd $inp } else { # delete it -- ie, do not copy it -- just update the counter # Do not use 'incr' just in case it = -1 set news_ncount [expr [news_count] - 1] } } else { puts $newfd $inp } incr next } close $fd close $newfd exec /bin/mv ${newsfile}~new $newsfile puthelp "NOTICE $nick :Deleted from the news file, thanks!" return 1 } # Read the channel news privately proc news_read {nick uhost hand arg} { global newsfile channel botnick news_biggest_ts set otherargs [lrange $arg 1 end] if {[string trim $otherargs] != ""} { set targ [scan $otherargs "%s " textarg] if {[string compare [string tolower $textarg] "all"] == 0} { if {[news_thereisnews]} { puthelp "NOTICE $nick :\002All news on $channel for [date] ...\002" news_print 0 $nick 0 0 puthelp "NOTICE $nick :\002--- end ---\002" } else { puthelp "NOTICE $nick :Sorry, there is no news at the moment!" } puthelp "NOTICE $nick :For help with other news commands, \002/msg \ $botnick HELP NEWS\002" return 1 } set havearg [scan $otherargs "%d" which] if {$havearg} { if {($which < 1) || ($which > [news_count])} { puthelp "NOTICE $nick :The item number must be between 1 and [news_count]" return 0 } else { news_print 0 $nick $which 0 return 1 } } } # By the time we are here, it must have no arguments - display only new ones! if {[news_thereisnews]} { set user_ts [news_get_ts $hand] if {$user_ts >= $news_biggest_ts} { puthelp "NOTICE $nick :Sorry, there is no new news at the moment!" puthelp "NOTICE $nick :To read all news, \002/msg $botnick NEWS READ ALL\002" } else { puthelp "NOTICE $nick :\002New news for $nick on $channel for [date] ...\002" news_print 0 $nick 0 $user_ts puthelp "NOTICE $nick :\002--- end ---\002" news_set_ts $hand $news_biggest_ts } } else { puthelp "NOTICE $nick :Sorry, there is no news at the moment!" } puthelp "NOTICE $nick :For help with other news commands, \002/msg $botnick HELP NEWS\002" return 1 } # Show the channel news publicly proc news_show {nick uhost hand arg} { global newsfile botnick channel news_lastseen set otherargs [lrange $arg 1 end] if {[string trim $otherargs] != ""} { set havearg [scan $otherargs "%d" which] if {$havearg} { if {($which < 1) || ($which > [news_count])} { puthelp "NOTICE $nick :The item number must be between 1 and [news_count]" return 0 } else { news_print 1 $channel $which 0 return 1 } } } # By the time we are here, it must have no arguments - display all if {([news_thereisnews]) && ([expr ([unixtime] - $news_lastseen) > 60])} { putserv "PRIVMSG $channel :\002NEWS on $channel for [date] ...\002 \(as requested by $nick\)" news_print 1 $channel 0 0 putserv "PRIVMSG $channel :\002--- end ---\002" putserv "PRIVMSG $channel :\002/msg $botnick HELP NEWS for instructions on how to use news\002" set news_lastseen [unixtime] } elseif {[news_thereisnews]} { puthelp "NOTICE $nick :News already displayed. I won\'t display more than once a minute." puthelp "NOTICE $nick :To see the news again privately, /msg $botnick NEWS READ" return 0 } else { puthelp "NOTICE $nick :Sorry, there is no news at the moment!" } return 1 } # Announce the availability of news to the channel proc news_announce {nick uhost hand arg} { if {[news_int_announce]} { puthelp "NOTICE $nick :Announced news availability to the channel." } else { puthelp "NOTICE $nick :No news to announce!" } return 1 } # Reset the counter associated with the news file. proc news_reset {nick uhost hand arg} { global news_ncount news_biggest_ts if {![matchattr $hand "m"]} { puthelp "NOTICE $nick :You don't have permission to reset the news file." return 0 } # Just reset the counter, don't touch the file set news_ncount -1 set news_biggest_ts 0 puthelp "NOTICE $nick :OK, reset." return 1 } # Erase all entries in the news file proc news_clear {nick uhost hand arg} { global newsfile news_ncount news_newnews if {![matchattr $hand "m"]} { puthelp "NOTICE $nick :You don't have permission to clear the news file." return 0 } set fd [open $newsfile w] puts $fd " " close $fd set news_ncount -1 set news_newnews 0 putserv "NOTICE $nick :Okay, news file cleared!" return 1 } # Check what news there is to be read proc news_check {nick uhost hand arg} { global botnick channel set unread [news_unread $hand] if {$unread} { putserv "NOTICE $nick :There is $channel news ($unread unread, [news_count] total). \002/msg $botnick NEWS READ\002" } retur1 } # Display some help proc news_help {nick uhost hand arg} { puthelp "NOTICE $nick :Usage: NEWS ADD <your news>" puthelp "NOTICE $nick : NEWS ERASE #" puthelp "NOTICE $nick : NEWS READ \[#\]" puthelp "NOTICE $nick : NEWS SHOW \[#\]" puthelp "NOTICE $nick : NEWS ANNOUNCE" if {[matchattr $hand "m"]} { puthelp "NOTICE $nick : NEWS RESET" puthelp "NOTICE $nick : NEWS CLEAR" } return 0 } # The main binding - does the dispatching of requests proc msg_news {nick uhost hand arg} { switch [string tolower [lindex $arg 0]] { "add" {set r [news_add $nick $uhost $hand $arg]} "new" {set r [news_add $nick $uhost $hand $arg]} "del" {set r [news_del $nick $uhost $hand $arg]} "delete" {set r [news_del $nick $uhost $hand $arg]} "erase" {set r [news_del $nick $uhost $hand $arg]} "read" {set r [news_read $nick $uhost $hand $arg]} "show" {set r [news_show $nick $uhost $hand $arg]} "ann" {set r [news_announce $nick $uhost $hand $arg]} "announce" {set r [news_announce $nick $uhost $hand $arg]} "check" {set r [news_check $nick $uhost $hand $arg]} "reset" {set r [news_reset $nick $uhost $hand $arg]} "clear" {set r [news_clear $nick $uhost $hand $arg]} default {set r [news_help $nick $uhost $hand $arg]} } return $r } putlog "NEWS script v0.4 loaded successfully!"